/*******************************************************************************
MPLAB Harmony Application Source File

Company:
Microchip Technology Inc.

File Name:
app.c

Summary:
This file contains the source code for the MPLAB Harmony application.

Description:
This file contains the source code for the MPLAB Harmony application. It
implements the logic of the application's state machine and it may call
API routines of other MPLAB Harmony modules in the system, such as drivers,
system services, and middleware. However, it does not call any of the
system interfaces (such as the "Initialize" and "Tasks" functions) of any of
the modules in the system or make any assumptions about when those functions
are called. That is the responsibility of the configuration-specific system
files.
*******************************************************************************/

// DOM-IGNORE-BEGIN
/*******************************************************************************
Copyright (c) 2013-2014 released Microchip Technology Inc. All rights reserved.

Microchip licenses to you the right to use, modify, copy and distribute
Software only when embedded on a Microchip microcontroller or digital signal
controller that is integrated into your product or third party product
(pursuant to the sublicense terms in the accompanying license agreement).

You should refer to the license agreement accompanying this Software for
additional information regarding your rights and obligations.

SOFTWARE AND DOCUMENTATION ARE PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF
MERCHANTABILITY, TITLE, NON-INFRINGEMENT AND FITNESS FOR A PARTICULAR PURPOSE.
IN NO EVENT SHALL MICROCHIP OR ITS LICENSORS BE LIABLE OR OBLIGATED UNDER
CONTRACT, NEGLIGENCE, STRICT LIABILITY, CONTRIBUTION, BREACH OF WARRANTY, OR
OTHER LEGAL EQUITABLE THEORY ANY DIRECT OR INDIRECT DAMAGES OR EXPENSES
INCLUDING BUT NOT LIMITED TO ANY INCIDENTAL, SPECIAL, INDIRECT, PUNITIVE OR
CONSEQUENTIAL DAMAGES, LOST PROFITS OR LOST DATA, COST OF PROCUREMENT OF
SUBSTITUTE GOODS, TECHNOLOGY, SERVICES, OR ANY CLAIMS BY THIRD PARTIES
(INCLUDING BUT NOT LIMITED TO ANY DEFENSE THEREOF), OR OTHER SIMILAR COSTS.
*******************************************************************************/
// DOM-IGNORE-END


// *****************************************************************************
// *****************************************************************************
// Section: Included Files
// *****************************************************************************
// *****************************************************************************

#include "app.h"

#include "stdio.h"


/* Macro defines USB internal DMA Buffer criteria*/

#define APP_MAKE_BUFFER_DMA_READY __attribute__((coherent)) __attribute__((aligned(16)))
//coherent: キャッシュ化不可、指定メモリ領域…KSEG1(カーネルセグメント1)
//aligned(16):16バイト単位で要メモリ確保
#define APP_READ_BUFFER_SIZE 512
uint8_t APP_MAKE_BUFFER_DMA_READY readBuffer[APP_READ_BUFFER_SIZE];



int delay_Clock = 200000000; //システムクロック:200MHz
char Buf[32];

int ix = 0;
char* str; //返信文字列へのポインタ for uint8_t APP_MAKE_BUFFER_DMA_READY
char* str2; //返信文字列へのポインタ2 for char

char Ready[] = "Are you Ready ?\r";
char Japan[] = "Japan\r";
char America[] = "America\r";
char UK[] = "U.K.\r";
char France[] = "France\r";
char IamFine[] = "I am fine\r";

char IamReady[] = "I am Ready";

//返信用 "Tokyo","Washington","London","Paris","Pardon ?"には、
//char と uint8_t APP_MAKE_BUFFER_DMA_READYの2つを定義
char Tokyo[] = "Tokyo";
char Washington[] = "Washington";
char London[] = "London";
char Paris[] = "Paris";
char Pardon[] = "Pardon ?";

uint8_t APP_MAKE_BUFFER_DMA_READY _IamReady[] = "I am Ready\r\n";
uint8_t APP_MAKE_BUFFER_DMA_READY _Tokyo[] = "Tokyo";
uint8_t APP_MAKE_BUFFER_DMA_READY _Washington[] = "Washington";
uint8_t APP_MAKE_BUFFER_DMA_READY _London[] = "London";
uint8_t APP_MAKE_BUFFER_DMA_READY _Paris[] = "Paris";
uint8_t APP_MAKE_BUFFER_DMA_READY _Pardon[] = "Pardon ?";


void delay_us(volatile unsigned int usec) //1μsec遅延
{
volatile int count;

count = (int)(delay_Clock/20000000)*usec;


do //実測 at 200MH (Clock=200000000)
{ //delay_us(1000):1000.4μsec delay_us(100):100.6μsec delay_us(10):10.5μsec  delay_us(1):1.5μsec
asm("NOP"); asm("NOP"); asm("NOP"); asm("NOP"); asm("NOP");asm("NOP");
asm("NOP"); asm("NOP"); asm("NOP"); asm("NOP"); asm("NOP"); asm("NOP");

count--;
}while(count != 0);
}

void delay_ms(volatile unsigned int msec) //1msec遅延
{
volatile unsigned int i; //実測:at200MH (Clock=200000000)//delay_ms(1): 1.0006msec delay_ms(100):100.04msec

for(i=0; i<msec; i++)
delay_us(1000);
}



// *****************************************************************************
/* Application Data

Summary:
Holds application data

Description:
This structure holds the application's data.

Remarks:
This structure should be initialized by the APP_Initialize function.

Application strings and buffers are be defined outside this structure.
*/

APP_DATA appData;


// *****************************************************************************
// *****************************************************************************
// Section: Application Callback Functions
// *****************************************************************************
// *****************************************************************************



/*******************************************************
* USB CDC Device Events - Application Event Handler
*******************************************************/

USB_DEVICE_CDC_EVENT_RESPONSE APP_USBDeviceCDCEventHandler
(
USB_DEVICE_CDC_INDEX index ,
USB_DEVICE_CDC_EVENT event ,
void * pData,
uintptr_t userData
)
{
APP_DATA * appDataObject;
appDataObject = (APP_DATA *)userData;
USB_CDC_CONTROL_LINE_STATE * controlLineStateData;

switch ( event )
{
case USB_DEVICE_CDC_EVENT_GET_LINE_CODING:

/* This means the host wants to know the current line
* coding. This is a control transfer request. Use the
* USB_DEVICE_ControlSend() function to send the data to
* host. */

USB_DEVICE_ControlSend(appDataObject->deviceHandle,
&appDataObject->getLineCodingData, sizeof(USB_CDC_LINE_CODING));

break;

case USB_DEVICE_CDC_EVENT_SET_LINE_CODING:

/* This means the host wants to set the line coding.
* This is a control transfer request. Use the
* USB_DEVICE_ControlReceive() function to receive the
* data from the host */

USB_DEVICE_ControlReceive(appDataObject->deviceHandle,
&appDataObject->setLineCodingData, sizeof(USB_CDC_LINE_CODING));

break;

case USB_DEVICE_CDC_EVENT_SET_CONTROL_LINE_STATE:

/* This means the host is setting the control line state.
* Read the control line state. We will accept this request
* for now. */

controlLineStateData = (USB_CDC_CONTROL_LINE_STATE *)pData;
appDataObject->controlLineStateData.dtr = controlLineStateData->dtr;
appDataObject->controlLineStateData.carrier = controlLineStateData->carrier;

USB_DEVICE_ControlStatus(appDataObject->deviceHandle, USB_DEVICE_CONTROL_STATUS_OK);

break;

case USB_DEVICE_CDC_EVENT_SEND_BREAK:

/* This means that the host is requesting that a break of the
* specified duration be sent. Read the break duration */

appDataObject->breakData = ((USB_DEVICE_CDC_EVENT_DATA_SEND_BREAK *)pData)->breakDuration;

/* Complete the control transfer by sending a ZLP */
USB_DEVICE_ControlStatus(appDataObject->deviceHandle, USB_DEVICE_CONTROL_STATUS_OK);

break;

case USB_DEVICE_CDC_EVENT_READ_COMPLETE:

/* This means that the host has sent some data*/
appDataObject->isReadComplete = true;
break;

case USB_DEVICE_CDC_EVENT_CONTROL_TRANSFER_DATA_RECEIVED:

/* The data stage of the last control transfer is
* complete. For now we accept all the data */

USB_DEVICE_ControlStatus(appDataObject->deviceHandle, USB_DEVICE_CONTROL_STATUS_OK);
break;

case USB_DEVICE_CDC_EVENT_CONTROL_TRANSFER_DATA_SENT:

/* This means the GET LINE CODING function data is valid. We dont
* do much with this data in this demo. */
break;

case USB_DEVICE_CDC_EVENT_WRITE_COMPLETE:

/* This means that the data write got completed. We can schedule
* the next read. */

appDataObject->isWriteComplete = true;
break;

default:
break;
}

return USB_DEVICE_CDC_EVENT_RESPONSE_NONE;
}

/***********************************************
* Application USB Device Layer Event Handler.
***********************************************/
void APP_USBDeviceEventHandler ( USB_DEVICE_EVENT event, void * eventData, uintptr_t context )
{
USB_DEVICE_EVENT_DATA_CONFIGURED *configuredEventData;

switch ( event )
{
case USB_DEVICE_EVENT_SOF:

/* This event is used for switch debounce. This flag is reset
* by the switch process routine. */
appData.sofEventHasOccurred = true;
break;

case USB_DEVICE_EVENT_RESET:

/* Update LED to show reset state */
// BSP_LEDOn ( APP_USB_LED_1 );
// BSP_LEDOn ( APP_USB_LED_2 );
// BSP_LEDOff ( APP_USB_LED_3 );

appData.isConfigured = false;

break;

case USB_DEVICE_EVENT_CONFIGURED:

/* Check the configuratio. We only support configuration 1 */
configuredEventData = (USB_DEVICE_EVENT_DATA_CONFIGURED*)eventData;
if ( configuredEventData->configurationValue == 1)
{
/* Update LED to show configured state */
// BSP_LEDOff ( APP_USB_LED_1 );
// BSP_LEDOff ( APP_USB_LED_2 );
// BSP_LEDOn ( APP_USB_LED_3 );

/* Register the CDC Device application event handler here.
* Note how the appData object pointer is passed as the
* user data */

USB_DEVICE_CDC_EventHandlerSet(USB_DEVICE_CDC_INDEX_0, APP_USBDeviceCDCEventHandler, (uintptr_t)&appData);

/* Mark that the device is now configured */
appData.isConfigured = true;

}
break;

case USB_DEVICE_EVENT_POWER_DETECTED:

/* VBUS was detected. We can attach the device */
USB_DEVICE_Attach(appData.deviceHandle);
break;

case USB_DEVICE_EVENT_POWER_REMOVED:

/* VBUS is not available any more. Detach the device. */
USB_DEVICE_Detach(appData.deviceHandle);
break;

case USB_DEVICE_EVENT_SUSPENDED:

/* Switch LED to show suspended state */
// BSP_LEDOff ( APP_USB_LED_1 );
// BSP_LEDOn ( APP_USB_LED_2 );
// BSP_LEDOn ( APP_USB_LED_3 );
break;

case USB_DEVICE_EVENT_RESUMED:
case USB_DEVICE_EVENT_ERROR:
default:
break;
}
}


/*****************************************************
* This function is called in every step of the
* application state machine.
*****************************************************/

bool APP_StateReset(void) //USBコネクタを抜き差しした場合などに機能する
{
/* This function returns true if the device
* was reset */

bool retVal;

if(appData.isConfigured == false)
{
appData.state = APP_STATE_WAIT_FOR_CONFIGURATION;
appData.readTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;
appData.writeTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;
appData.isReadComplete = true;
appData.isWriteComplete = true;
retVal = true;
}
else
{
retVal = false;
}

return(retVal);
}


// *****************************************************************************
// *****************************************************************************
// Section: Application Initialization and State Machine Functions
// *****************************************************************************
// *****************************************************************************

/*******************************************************************************
Function:
void APP_Initialize ( void )

Remarks:
See prototype in app.h.
*/

void APP_Initialize ( void )
{
/* Place the App state machine in its initial state. */
appData.state = APP_STATE_INIT;

/* Device Layer Handle */
appData.deviceHandle = USB_DEVICE_HANDLE_INVALID ;

/* Device configured status */
appData.isConfigured = false;

/* Initial get line coding state */
appData.getLineCodingData.dwDTERate = 9600;
appData.getLineCodingData.bParityType = 0;
appData.getLineCodingData.bParityType = 0;
appData.getLineCodingData.bDataBits = 8;

/* Read Transfer Handle */
appData.readTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;

/* Write Transfer Handle */
appData.writeTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;

/* Intialize the read complete flag */
appData.isReadComplete = true;

/*Initialize the write complete flag*/
appData.isWriteComplete = true;

/* Reset other flags */
appData.sofEventHasOccurred = false;

/* Set up the read buffer */
appData.readBuffer = &readBuffer[0];


lcd_ACM1602_init_i2c(); //I2Cインターフェース式液晶初期化
lcd_ACM1602_cmd_i2c(0x0C); //カーソル:0FF、ブリンク:0FF


lcd_ACM1602_cmd_i2c(0x80); //1行目の先頭へ
lcd_ACM1602_str_i2c("USB device CDC ");

delay_ms(1000);


lcd_ACM1602_cmd_i2c(0xC0); //2行目の先頭へ
lcd_ACM1602_str_i2c(" Start !! ");


delay_ms(2000);



}


/******************************************************************************
Function:
void APP_Tasks ( void )

Remarks:
See prototype in app.h.
*/

void APP_Tasks (void )
{
/* Update the application state machine based
* on the current state */

switch(appData.state)
{
case APP_STATE_INIT:

/* Open the device layer */
appData.deviceHandle = USB_DEVICE_Open( USB_DEVICE_INDEX_0, DRV_IO_INTENT_READWRITE );

if(appData.deviceHandle != USB_DEVICE_HANDLE_INVALID)
{
/* Register a callback with device layer to get event notification (for end point 0) */
USB_DEVICE_EventHandlerSet(appData.deviceHandle, APP_USBDeviceEventHandler, 0);

appData.state = APP_STATE_WAIT_FOR_CONFIGURATION;
}
else
{
/* The Device Layer is not ready to be opened. We should try
* again later. */
}

break;

case APP_STATE_WAIT_FOR_CONFIGURATION:

/* Check if the device was configured */
if(appData.isConfigured)
{
/* If the device is configured then lets start reading */
appData.state = APP_STATE_SCHEDULE_READ;
}
break;

case APP_STATE_SCHEDULE_READ:

if(APP_StateReset())
{
break;
}

/* If a read is complete, then schedule a read
* else wait for the current read to complete */

appData.state = APP_STATE_WAIT_FOR_READ_COMPLETE;
if(appData.isReadComplete == true)
{
appData.isReadComplete = false;
appData.readTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;

USB_DEVICE_CDC_Read (USB_DEVICE_CDC_INDEX_0,
&appData.readTransferHandle, appData.readBuffer,
APP_READ_BUFFER_SIZE);

if(appData.readTransferHandle == USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID)
{
appData.state = APP_STATE_ERROR;
break;
}
}

break;

case APP_STATE_WAIT_FOR_READ_COMPLETE:

if(APP_StateReset())
{
break;
}


/* Check if a character was received or a switch was pressed.
* The isReadComplete flag gets updated in the CDC event handler. */

if(appData.isReadComplete)
{
appData.state = APP_STATE_SCHEDULE_WRITE;
}

break;


case APP_STATE_SCHEDULE_WRITE:

if(APP_StateReset()) //リセットが発生した場合
{
break; //(送信処理をしないで)このステート終了
}

/* Setup the write */

appData.writeTransferHandle = USB_DEVICE_CDC_TRANSFER_HANDLE_INVALID;
appData.isWriteComplete = false;
appData.state = APP_STATE_WAIT_FOR_WRITE_COMPLETE;


if(strcmp(readBuffer,Ready) == 0)
{ //下記2つの文字列は分けて処理のこと
str = _IamReady; //返信用文字列
str2 = &IamReady[0]; //表示用文字列
}
else if(strcmp(readBuffer,Japan) == 0) //Japanの場合 → Tokyo
{
str = _Tokyo;
str2 = &Tokyo[0];
}
else if(strcmp(readBuffer,America) == 0) //Americaの場合 → Washinton
{
str = _Washington;
str2 = &Washington[0];
}
else if(strcmp(readBuffer,UK) == 0) // U.K.の場合 → London
{
str = _London;
str2 = &London[0];
}
else if(strcmp(readBuffer,France) == 0) //Franceの場合 → Paris
{
str = _Paris;
str2 = &Paris[0];
}
else //その他の場合 → Pardon ?
{
str = _Pardon;
str2 = &Pardon[0];
}


//I2C制御キャラクタ液晶表示
lcd_ACM1602_cmd_i2c(0x80); //1行目の先頭へ
sprintf(Buf,"%s ",readBuffer);//
lcd_ACM1602_str_i2c(Buf); //液晶表示

for(ix = 0; ix < 20; ix++)Buf[ix] = 0x20; //バッファークリア //必須


lcd_ACM1602_cmd_i2c(0xC0); //2行目の先頭へ
sprintf(Buf,"%s ",str2); //
lcd_ACM1602_str_i2c(Buf); // 開始メッセージ1行目表示

for(ix = 0; ix < 20; ix++)readBuffer[ix] = '\0'; //バッファークリア //必須


//文字列を返信
appData.readBuffer[0] = appData.readBuffer[0] + 1;
USB_DEVICE_CDC_Write(USB_DEVICE_CDC_INDEX_0,
&appData.writeTransferHandle,
str, 12, //12バイト: 最大送信データサイズ //12バイト: _IamReady[] = "I am Ready\r\n";
//strのサイズが12バイトより小さい場合 unknownデータが送られるので要注意
USB_DEVICE_CDC_TRANSFER_FLAGS_DATA_COMPLETE);

break;

case APP_STATE_WAIT_FOR_WRITE_COMPLETE:

if(APP_StateReset())
{
break; //
}

/* Check if a character was sent. The isWriteComplete
* flag gets updated in the CDC event handler */

if(appData.isWriteComplete == true)
{
appData.state = APP_STATE_SCHEDULE_READ;
}

break;

case APP_STATE_ERROR:
break;
default:
break;
}

}

/*******************************************************************************
End of File
*/